I have Swift code that accepts dragging a lowercase string from the TextEdit app.
On the first drop, the text is accepted as is but, on the second drop, the first words are capitalised.
Any ideas?
Post
Replies
Boosts
Views
Activity
Well, it's all in the title.
I want to make a URL history list available on the dock item of my macOS app.
I found NSDockTilePlugin but nothing on how to implement it.
Can anyone help?
Can anyone tell me how to retrieve all Finder tags, including user ones that may have been synced from iCloud?
I have an app where each item in an NSTableView contains an NSMetadataQuery, to listen for updates to image files that contain the keyword associated with the item.
If I write the keywords to, RAW files, everything works as expected and the number label in the row gets updated correctly.
However, if I write the keywords to XMP sidecar files, the NSMetadataQueryDidFinishGathering callback isn't called enough times and the count of files affected isn't updated correctly.
What is more, the number returned on the final call to the callback seems to vary depending on the time of day, weather, etc
Here is the code that I have used to create the predicate for finding keywords in XMP files…
let xmpFileTypePredicate = NSPredicate(fromMetadataQueryString: "kMDItemKind = \"XMP sidecar*\"cdwt")!
let xmpKeywordPredicate = NSPredicate(fromMetadataQueryString: "kMDItemTextContent = \"<rdf:li>\(keyword)</rdf:li>\"cdw")!
let xmpKeywordsMetadataQueryPredicate = NSCompoundPredicate(andPredicateWithSubpredicates: [xmpFileTypePredicate, xmpKeywordPredicate])
Can anyone give me any ideas?
Is there any way to retrieve the list of Finder Tags now that the SyncedPreferences option is no longer available for Monterey?
I have a long established app that uses a threadsafe var to protect an array of URLs from concurrency races, etc…
class ThreadsafeVar<T>
{
private var value = Optional<T>.none
func callAsFunction() -> T?
{
queue.sync(flags: .barrier) { [unowned self] in value }
}
func set(_ value: T?)
{
queue.async(flags: .barrier) { [unowned self] in self.value = value }
}
}
What I want to do is substitute an Actor, which I understand will be intrinsically threadsafe. So, I wrote one like this…
actor UrlManager
{
private var value = [URL]()
func callAsFunction() -> [URL]
{
value
}
func set(_ value: [URL])
{
self.value = value
}
}
The calling code is in one of four classes, all of which implement a protocol, which contains a read-write var…
protocol FolderProtocol
{
…
var fileURLs: [URL] { get set }
}
In this particular class, the var is implemented as read-only…
private var urls = ThreadsafeVar<[URL]>()
var fileURLs: [URL]
{
get
{
urls()?.sorted(using: .finder, pathComponent: .fullPath) ?? []
}
set { }
}
… the private var being updated from a metadata updating notification handler.
Calling the set(_:[URL]) method on the actor poses no problem…
Task { await urls.set(foundURLs) }
But, no matter what I try - continuation, try, await, etc, I don't seem to be able to read back the array of URLs from the actor's callAsFunction() method.
The fileURLs var is called on the main thread in response to NSCollectionView delegate methods.
Other implementing classes don't need the concurrency management as they are fixed once and never changed.
Can anyone help me with this one?
Greetings allI have an app whose development language is French, which is also its base localization. I also localise it to English.The app makes use of both location services and the calendar.I have created an InfoPlist.strings file and localized both NSCalendarsUsageDescription and NSLocationWhenInUseUsageDescription.My problem is that, although the Base localization (French) of NSCalendarsUsageDescription is correctly used when requesting permission, I can only ever see the French text for NSLocationWhenInUseUsageDescription if I remove the key from the English version of InfoPlist.strings.Many thanks for any help you can give me.
I have an NSCollectionView which is showing thumbnails of images and I have just switched to using QLThumbnailGenerator to fetch them.
There are over 6,000 possible images that can be viewed and, if I scroll too fast, I start to get the wrong thumbnails returned from the generator.
Is this a bug, or is there something I can do to fix this?
Here is the code that is written inside the NSCollectionViewItem derived class…
var request: QLThumbnailGenerator.Request?
func loadImage()
{
if imageView?.image != NSImage(imageLiteralResourceName: "Placeholder")
{
return
}
request = QLThumbnailGenerator.Request(fileAt: url!, size: imageSize, scale: 1.0, representationTypes: [.lowQualityThumbnail])
QLThumbnailGenerator.shared.generateBestRepresentation(for: request!)
{
(thumbnail: QLThumbnailRepresentation?, error: Error?) -> Void in
if let request = self.request
{
QLThumbnailGenerator.shared.cancel(request)
}
DispatchQueue.main.async
{
[unowned self] in
if self.imageView?.image != NSImage(imageLiteralResourceName: "Placeholder")
{
return
}
let transition = CATransition()
transition.timingFunction = CAMediaTimingFunction(name: CAMediaTimingFunctionName.easeInEaseOut)
transition.duration = 0.3
imageView?.layer?.add(transition, forKey: nil)
imageView?.image = thumbnail?.nsImage
…
Is it possible to change the background colour of an RTF file shown in NSHelpManager to match dark mode?
I need to monitor a file for when its metadata changes.
I'm using the following code I found.
final class FileMonitor
{
let url: URL
var fileHandle: FileHandle?
var source: DispatchSourceFileSystemObject?
var didReceiveChanges: (() - Void)?
func process(event: DispatchSource.FileSystemEvent)
{
print(String(describing: event))
didReceiveChanges?()
}
func eventHandler()
{
let event = self.source!.data
process(event: event)
createDispatchSource()
}
func createDispatchSource()
{
self.fileHandle = try? FileHandle(forReadingFrom: url)
source = DispatchSource.makeFileSystemObjectSource(fileDescriptor: fileHandle!.fileDescriptor, eventMask: [.attrib], queue: DispatchQueue.main)
source!.setEventHandler(handler: eventHandler)
source!.setCancelHandler
{
self.fileHandle!.closeFile()
}
source!.resume()
}
init(url: URL) throws
{
self.url = url
createDispatchSource()
}
deinit
{
source!.cancel()
}
}
The problem is that I am getting nearly 7,000 notifications every time a change is detected, which is causing the CPU to get rather warm.
This subject seems to be very poorly documented and after days searching, I don't seem to be able to make things any better.
Any ideas please?
Joanna
I have a macOS storyboard with a main Window/View Controller and a couple of "accessory" view controllers that are shown via segues from buttons.
The problem is, even having set the restoration ID of both controllers in the storyboard, that they always start at the same position, even if they have been moved.
Can anyone help me to get them to reopen in the same place they were when they were closed?
I have just spent an amusing hour or two trying to track down why my app couldn't find a certain subfolder of a folder I had already enumerated.
In the end I found that subpathsOfDirectory wasn't finding some folders, whereas, in the end, I found using subpaths correctly listed all subfolders.
Any ideas why?
I have created a wrapped NSTextField to allow for keystroke validation.
class ValidatableTextField: NSTextField, NSTextViewDelegate
{
var validationClosure: ((ValidatableTextField, NSRange, String?) -> Bool)?
func textView(_ textView: NSTextView, shouldChangeTextIn affectedCharRange: NSRange, replacementString: String?) -> Bool
{
return validationClosure?(self, affectedCharRange, replacementString) ?? true
}
}
struct CurrencyTextField<rootT : AnyObject, valueT> : NSViewRepresentable
{
private var object: rootT?
private var keyPath: ReferenceWritableKeyPath<rootT, valueT>?
func makeCoordinator() -> Coordinator
{
return Coordinator(object, keyPath: keyPath)
}
init(object: rootT?, keyPath: ReferenceWritableKeyPath<rootT, valueT>?)
{
self.object = object
self.keyPath = keyPath
}
func makeNSView(context: Context) -> ValidatableTextField
{
let textField = ValidatableTextField()
textField.validationClosure = context.coordinator.validationClosure
return textField
}
func updateNSView(_ nsView: ValidatableTextField, context: Context)
{
guard let object = object,
let keyPath = keyPath else
{
return
}
nsView.stringValue = "\(object[keyPath: keyPath])"
}
class Coordinator : NSObject
{
var object: rootT?
var keyPath: ReferenceWritableKeyPath<rootT, valueT>?
init(_ object: rootT?, keyPath: ReferenceWritableKeyPath<rootT, valueT>?)
{
self.object = object
self.keyPath = keyPath
}
lazy var validationClosure: ((ValidatableTextField, NSRange, String?) -> Bool) =
{
textField, range, replacementString in
if replacementString!.isEmpty
{
return true
}
return replacementString?.contains(where:
{
(char: Character) -> Bool in
return char.isNumber || String(char) == Locale.autoupdatingCurrent.decimalSeparator
}) ?? false
}
}
}
struct ContentView: View
{
@ObservedObject var person: Person
var body: some View
{
VStack
{
Spacer()
Text(person.name)
HStack
{
Spacer(minLength: 100)
CurrencyTextField<Person, Double>(object: person, keyPath: \.bankBalance)
Text("\(person.bankBalance)")
Spacer(minLength: 100)
}
Button(action:
{
self.person.bankBalance = 65.43
})
{
Text("Change bank balance")
}
Spacer()
}
}
}
My problem is that I am setting the bankBalance property on the Person and the Text next to the TextField is being updated but my CurrencyTextField is not.
I may have 30 years of programming experience, including quite a few with Swift but SwiftUI, beyond the simplest of examples, is proving a bit difficult. What am I missing here please?
I have some code that calls an external library asynchronously to read EXIF keywords from image files :
	func readKeywords()
	{
		keywordsForSelectedItems.removeAll()
		
		let group: DispatchGroup = .init()
		
		let queue: DispatchQueue = .init(label: "com.keywords")
		
		collectionView.selectionIndexPaths.forEach
		{
			indexPath in
			
			let item = self.collectionView.item(at: indexPath) as! CollectionViewItem
			
			let url = item.url!
			
			group.enter()
			
			queue.async(group: group)
			{
				do
				{
					try ExifReader.keywords(for: url)
					{
						keywords in
						
						if let keywords = keywords,
							 keywords.count > 0
						{
							self.keywordsForSelectedItems.append(keywords)
						}
						
						group.leave()
					}
				}
				catch
				{
					print(error.localizedDescription)
				}
			}
		}
		group.notify(queue: .main)
		{
			let firstValue: [String]? = self.keywordsForSelectedItems.first
			{
				keywords in
				
				keywords.count > 0
			}
				
			if firstValue != nil
			{
				let keywords = self.keywordsForSelectedItems.reduce(Set(firstValue!))
				{
					result, item in
					
					result.union(item)
				}
				
				self.tokenView.objectValue = keywords.sorted()
			}
		}
	}
Before I added this code, everything worked fine and images were displayed perfectly in a CollectionView.
But, since adding this code, as soon as this code has been called, the CPU fan starts to run, anywhere up to 1000% at times and it can block other asynchronous processes that normally load images in the background.
Anyone got any ideas?